home *** CD-ROM | disk | FTP | other *** search
/ Chip 2007 January, February, March & April / Chip-Cover-CD-2007-02.iso / Pakiet internetowy / Przegladarki internetowe / Mozilla Seamonkey 1.0.5 pl / seamonkey-1.0.5.pl-PL.win32.installer.exe / VENKMAN.XPI / bin / chrome / venkman.jar / content / venkman / command-manager.js next >
Encoding:
JavaScript  |  2004-07-19  |  23.8 KB  |  837 lines

  1. /* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 4 -*-
  2.  *
  3.  * ***** BEGIN LICENSE BLOCK *****
  4.  * Version: MPL 1.1/GPL 2.0/LGPL 2.1
  5.  *
  6.  * The contents of this file are subject to the Mozilla Public License Version
  7.  * 1.1 (the "License"); you may not use this file except in compliance with
  8.  * the License. You may obtain a copy of the License at
  9.  * http://www.mozilla.org/MPL/
  10.  *
  11.  * Software distributed under the License is distributed on an "AS IS" basis,
  12.  * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
  13.  * for the specific language governing rights and limitations under the
  14.  * License.
  15.  *
  16.  * The Original Code is JSIRC Library.
  17.  *
  18.  * The Initial Developer of the Original Code is
  19.  * New Dimensions Consulting, Inc.
  20.  * Portions created by the Initial Developer are Copyright (C) 1999
  21.  * the Initial Developer. All Rights Reserved.
  22.  *
  23.  * Contributor(s):
  24.  *   Robert Ginda, rginda@ndcico.com, original author
  25.  *
  26.  * Alternatively, the contents of this file may be used under the terms of
  27.  * either the GNU General Public License Version 2 or later (the "GPL"), or
  28.  * the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
  29.  * in which case the provisions of the GPL or the LGPL are applicable instead
  30.  * of those above. If you wish to allow use of your version of this file only
  31.  * under the terms of either the GPL or the LGPL, and not to allow others to
  32.  * use your version of this file under the terms of the MPL, indicate your
  33.  * decision by deleting the provisions above and replace them with the notice
  34.  * and other provisions required by the GPL or the LGPL. If you do not delete
  35.  * the provisions above, a recipient may use your version of this file under
  36.  * the terms of any one of the MPL, the GPL or the LGPL.
  37.  *
  38.  * ***** END LICENSE BLOCK ***** */
  39.  
  40. function getAccessKey (str)
  41. {
  42.     var i = str.indexOf("&");
  43.     if (i == -1)
  44.         return "";
  45.     return str[i + 1];
  46. }
  47.  
  48. function CommandRecord (name, func, usage, help, label, flags, keystr, tip)
  49. {
  50.     this.name = name;
  51.     this.func = func;
  52.     this._usage = usage;
  53.     this.scanUsage();
  54.     this.help = help;
  55.     this.label = label ? label : name;
  56.     this.labelstr = label.replace ("&", "");
  57.     this.tip = tip;
  58.     this.flags = flags;
  59.     this._enabled = true;
  60.     this.keyNodes = new Array();
  61.     this.keystr = keystr;
  62.     this.uiElements = new Array();
  63. }
  64.  
  65. CommandRecord.prototype.__defineGetter__ ("enabled", cr_getenable);
  66. function cr_getenable ()
  67. {
  68.     return this._enabled;
  69. }
  70.  
  71. CommandRecord.prototype.__defineSetter__ ("enabled", cr_setenable);
  72. function cr_setenable (state)
  73. {
  74.     for (var i = 0; i < this.uiElements.length; ++i)
  75.     {
  76.         if (state)
  77.             this.uiElements[i].removeAttribute ("disabled");
  78.         else
  79.             this.uiElements[i].setAttribute ("disabled", "true");
  80.     }
  81.     return (this._enabled = state);
  82. }
  83.  
  84. CommandRecord.prototype.__defineSetter__ ("usage", cr_setusage);
  85. function cr_setusage (usage)
  86. {
  87.     this._usage = usage;
  88.     this.scanUsage();
  89. }
  90.  
  91. CommandRecord.prototype.__defineGetter__ ("usage", cr_getusage);
  92. function cr_getusage()
  93. {
  94.     return this._usage;
  95. }
  96.  
  97. /**
  98.  * Internal use only.
  99.  *
  100.  * Scans the argument spec, in the format "<a1> <a2> [<o1> <o2>]", into an
  101.  * array of strings.
  102.  */
  103. CommandRecord.prototype.scanUsage =
  104. function cr_scanusage()
  105. {
  106.     var spec = this._usage;
  107.     var currentName = "";
  108.     var inName = false;
  109.     var len = spec.length;
  110.     var capNext = false;
  111.     
  112.     this._usage = spec;
  113.     this.argNames = new Array();
  114.  
  115.     for (var i = 0; i < len; ++i)
  116.     {
  117.         switch (spec[i])
  118.         {
  119.             case '[':
  120.                 this.argNames.push (":");
  121.                 break;
  122.                 
  123.             case '<':
  124.                 inName = true;
  125.                 break;
  126.                 
  127.             case '-':
  128.                 capNext = true;
  129.                 break;
  130.                 
  131.             case '>':
  132.                 inName = false;
  133.                 this.argNames.push (currentName);
  134.                 currentName = "";
  135.                 capNext = false;
  136.                 break;
  137.                 
  138.             default:
  139.                 if (inName)
  140.                     currentName += capNext ? spec[i].toUpperCase() : spec[i];
  141.                 capNext = false;
  142.                 break;
  143.         }
  144.     }
  145. }
  146.  
  147. /**
  148.  * Returns the command formatted for presentation as part of online
  149.  * documentation.
  150.  */
  151. CommandRecord.prototype.getDocumentation =
  152. function cr_getdocs(flagFormatter)
  153.  
  154. {
  155.     var str;
  156.     str  = getMsg(MSN_DOC_COMMANDLABEL,
  157.                   [this.label.replace("&", ""), this.name]) + "\n";
  158.     str += getMsg(MSN_DOC_KEY, this.keystr ? this.keystr : MSG_VAL_NA) + "\n";
  159.     str += getMsg(MSN_DOC_SYNTAX, [this.name, this.usage]) + "\n";
  160.     str += MSG_DOC_NOTES + "\n";
  161.     str += (flagFormatter ? flagFormatter(this.flags) : this.flags) + "\n\n";
  162.     str += MSG_DOC_DESCRIPTION + "\n";
  163.     str += wrapText(this.help, 75) + "\n";
  164.     return str;
  165. }
  166.  
  167. CommandRecord.prototype.argNames = new Array();
  168.  
  169. function CommandManager (defaultBundle)
  170. {
  171.     this.commands = new Object();
  172.     this.defaultBundle = defaultBundle;
  173. }
  174.  
  175. CommandManager.prototype.defineCommands =
  176. function cmgr_defcmds (cmdary)
  177. {
  178.     var len = cmdary.length;
  179.     var commands = new Object();
  180.     var bundle =  ("stringBundle" in cmdary ?
  181.                    cmdary.stringBundle : 
  182.                    this.defaultBundle);
  183.     
  184.     for (var i = 0; i < len; ++i)
  185.     {
  186.         var name  = cmdary[i][0];
  187.         var func  = cmdary[i][1];
  188.         var flags = cmdary[i][2];
  189.         var usage = getMsgFrom(bundle, "cmd." + name + ".params", null, "");
  190.  
  191.         var helpDefault;
  192.         var labelDefault = name;
  193.         var aliasFor;
  194.         if (flags & CMD_NO_HELP)
  195.             helpDefault = MSG_NO_HELP;
  196.  
  197.         if (typeof func == "string")
  198.         {
  199.             var ary = func.match(/(\S+)/);
  200.             if (ary)
  201.                 aliasFor = ary[1];
  202.             else
  203.                 aliasFor = null;
  204.             helpDefault = getMsg (MSN_DEFAULT_ALIAS_HELP, func); 
  205.             if (aliasFor)
  206.                 labelDefault = getMsgFrom (bundle, "cmd." + aliasFor + ".label",
  207.                                            null, name);
  208.         }
  209.  
  210.         var label = getMsgFrom(bundle, "cmd." + name + ".label", null,
  211.                                labelDefault);
  212.         var help  = getMsgFrom(bundle, "cmd." + name + ".help", null,
  213.                                helpDefault);
  214.         var keystr = getMsgFrom (bundle, "cmd." + name + ".key", null, "");
  215.         var tip = getMsgFrom (bundle, "cmd." + name + ".tip", null, "");
  216.         var command = new CommandRecord (name, func, usage, help, label, flags,
  217.                                          keystr, tip);
  218.         if (aliasFor)
  219.             command.aliasFor = aliasFor;
  220.         this.addCommand(command);
  221.         commands[name] = command;
  222.     }
  223.  
  224.     return commands;
  225. }
  226.  
  227. CommandManager.prototype.installKeys =
  228. function cmgr_instkeys (document, commands)
  229. {
  230.     var parentElem = document.getElementById("dynamic-keys");
  231.     if (!parentElem)
  232.     {
  233.         parentElem = document.createElement("keyset");
  234.         parentElem.setAttribute ("id", "dynamic-keys");
  235.         document.firstChild.appendChild (parentElem);
  236.     }
  237.  
  238.     if (!commands)
  239.         commands = this.commands;
  240.     
  241.     for (var c in commands)
  242.         this.installKey (parentElem, commands[c]);
  243. }
  244.  
  245. /**
  246.  * Create a <key> node relative to a DOM node.  Usually called once per command,
  247.  * per document, so that accelerator keys work in all application windows.
  248.  *
  249.  * @param parentElem  A reference to the DOM node which should contain the new
  250.  *                    <key> node.
  251.  * @param command     reference to the CommandRecord to install.
  252.  */
  253. CommandManager.prototype.installKey =
  254. function cmgr_instkey (parentElem, command)
  255. {
  256.     if (!command.keystr)
  257.         return;
  258.  
  259.     var ary = command.keystr.match (/(.*\s)?([\S]+)$/);
  260.     if (!ASSERT(ary, "couldn't parse key string ``" + command.keystr +
  261.                 "'' for command ``" + command.name + "''"))
  262.     {
  263.         return;
  264.     }
  265.     
  266.     var key = document.createElement ("key");
  267.     key.setAttribute ("id", "key:" + command.name);
  268.     key.setAttribute ("oncommand", "dispatch('" + command.name +
  269.                       "', {isInteractive: true});");
  270.     if (ary[1])
  271.         key.setAttribute ("modifiers", ary[1]);
  272.     if (ary[2].indexOf("VK_") == 0)
  273.         key.setAttribute ("keycode", ary[2]);
  274.     else
  275.         key.setAttribute ("key", ary[2]);
  276.     
  277.     parentElem.appendChild(key);
  278.     command.keyNodes.push(key);
  279. }
  280.  
  281. CommandManager.prototype.uninstallKeys =
  282. function cmgr_uninstkeys (commands)
  283. {
  284.     if (!commands)
  285.         commands = this.commands;
  286.     
  287.     for (var c in commands)
  288.         this.uninstallKey (commands[c]);
  289. }
  290.  
  291. CommandManager.prototype.uninstallKey =
  292. function cmgr_uninstkey (command)
  293. {
  294.     for (var i in command.keyNodes)
  295.     {
  296.         try
  297.         {
  298.             /* document may no longer exist in a useful state. */
  299.             command.keyNodes[i].parentNode.removeChild(command.keyNodes[i]);
  300.         }
  301.         catch (ex)
  302.         {
  303.             dd ("*** caught exception uninstalling key node: " + ex);
  304.         }
  305.     }
  306. }
  307.  
  308. /**
  309.  * Register a new command with the manager.
  310.  */
  311. CommandManager.prototype.addCommand =
  312. function cmgr_add (command)
  313. {
  314.     this.commands[command.name] = command;
  315. }
  316.  
  317. CommandManager.prototype.removeCommands = 
  318. function cmgr_removes (commands)
  319. {
  320.     for (var c in commands)
  321.         this.removeCommand(commands[c]);
  322. }
  323.  
  324. CommandManager.prototype.removeCommand = 
  325. function cmgr_remove (command)
  326. {
  327.     delete this.commands[command.name];
  328. }
  329.  
  330. /**
  331.  * Register a hook for a particular command name.  |id| is a human readable
  332.  * identifier for the hook, and can be used to unregister the hook.  If you
  333.  * re-use a hook id, the previous hook function will be replaced.
  334.  * If |before| is |true|, the hook will be called *before* the command executes,
  335.  * if |before| is |false|, or not specified, the hook will be called *after*
  336.  * the command executes.
  337.  */
  338. CommandManager.prototype.addHook =
  339. function cmgr_hook (commandName, func, id, before)
  340. {
  341.     if (!ASSERT(commandName in this.commands,
  342.                 "Unknown command '" + commandName + "'"))
  343.     {
  344.         return;
  345.     }
  346.     
  347.     var command = this.commands[commandName];
  348.     
  349.     if (before)
  350.     {
  351.         if (!("beforeHooks" in command))
  352.             command.beforeHooks = new Object();
  353.         command.beforeHooks[id] = func;
  354.     }
  355.     else
  356.     {
  357.         if (!("afterHooks" in command))
  358.             command.afterHooks = new Object();
  359.         command.afterHooks[id] = func;
  360.     }
  361. }
  362.  
  363. CommandManager.prototype.addHooks =
  364. function cmgr_hooks (hooks, prefix)
  365. {
  366.     if (!prefix)
  367.         prefix = "";
  368.     
  369.     for (var h in hooks)
  370.     {
  371.         this.addHook(h, hooks[h], prefix + ":" + h, 
  372.                      ("_before" in hooks[h]) ? hooks[h]._before : false);
  373.     }
  374. }
  375.  
  376. CommandManager.prototype.removeHooks =
  377. function cmgr_remhooks (hooks, prefix)
  378. {
  379.     if (!prefix)
  380.         prefix = "";
  381.     
  382.     for (var h in hooks)
  383.     {
  384.         this.removeHook(h, prefix + ":" + h, 
  385.                         ("before" in hooks[h]) ? hooks[h].before : false);
  386.     }
  387. }
  388.  
  389. CommandManager.prototype.removeHook =
  390. function cmgr_unhook (commandName, id, before)
  391. {
  392.     var command = this.commands[commandName];
  393.  
  394.     if (before)
  395.         delete command.beforeHooks[id];
  396.     else
  397.         delete command.afterHooks[id];
  398. }
  399.  
  400. /**
  401.  * Return an array of all CommandRecords which start with the string
  402.  * |partialName|, sorted by |label| property.
  403.  *
  404.  * @param   partialName Prefix to search for.
  405.  * @param   flags       logical ANDed with command flags.
  406.  * @returns array       Array of matching commands, sorted by |label| property.
  407.  */
  408. CommandManager.prototype.list =
  409. function cmgr_list (partialName, flags)
  410. {
  411.     /* returns array of command objects which look like |partialName|, or
  412.      * all commands if |partialName| is not specified */
  413.     function compare (a, b)
  414.     {
  415.         a = a.labelstr.toLowerCase();
  416.         b = b.labelstr.toLowerCase();
  417.  
  418.         if (a == b)
  419.             return 0;
  420.  
  421.         if (a > b)
  422.             return 1;
  423.  
  424.         return -1;
  425.     }
  426.  
  427.     var ary = new Array();
  428.     var commandNames = keys(this.commands);
  429.  
  430.     /* a command named "eval" wouldn't show up in the result of keys() because
  431.      * eval is not-enumerable, even if overwritten. */
  432.     if ("eval" in this.commands && typeof this.commands.eval == "object")
  433.         commandNames.push ("eval");
  434.  
  435.     for (var i in commandNames)
  436.     {
  437.         var name = commandNames[i];
  438.         if (!flags || (this.commands[name].flags & flags))
  439.         {
  440.             if (!partialName ||
  441.                 this.commands[name].name.indexOf(partialName) == 0)
  442.             {
  443.                 if (partialName && 
  444.                     partialName.length == this.commands[name].name.length)
  445.                 {
  446.                     /* exact match */
  447.                     return [this.commands[name]];
  448.                 }
  449.                 else
  450.                 {
  451.                     ary.push (this.commands[name]);
  452.                 }
  453.             }
  454.         }
  455.     }
  456.  
  457.     ary.sort(compare);
  458.     return ary;
  459. }
  460.  
  461. /**
  462.  * Return a sorted array of the command names which start with the string
  463.  * |partialName|.
  464.  *
  465.  * @param   partialName Prefix to search for.
  466.  * @param   flags       logical ANDed with command flags.
  467.  * @returns array       Sorted Array of matching command names.
  468.  */
  469. CommandManager.prototype.listNames =
  470. function cmgr_listnames (partialName, flags)
  471. {
  472.     var cmds = this.list(partialName, flags);
  473.     var cmdNames = new Array();
  474.     
  475.     for (var c in cmds)
  476.         cmdNames.push (cmds[c].name);
  477.  
  478.     cmdNames.sort();
  479.     return cmdNames;
  480. }
  481.  
  482. /**
  483.  * Internal use only.
  484.  *
  485.  * Called to parse the arguments stored in |e.inputData|, as properties of |e|,
  486.  * for the CommandRecord stored on |e.command|.
  487.  *
  488.  * @params e  Event object to be processed.
  489.  */
  490. CommandManager.prototype.parseArguments =
  491. function cmgr_parseargs (e)
  492. {
  493.     var rv = this.parseArgumentsRaw(e);
  494.     //dd("parseArguments '" + e.command.usage + "' " + 
  495.     //   (rv ? "passed" : "failed") + "\n" + dumpObjectTree(e));
  496.     delete e.currentArgIndex;
  497.     return rv;
  498. }
  499.  
  500. /**
  501.  * Internal use only.
  502.  *
  503.  * Don't call parseArgumentsRaw directly, use parseArguments instead.
  504.  *
  505.  * Parses the arguments stored in the |inputData| property of the event object,
  506.  * according to the format specified by the |command| property.
  507.  *
  508.  * On success this method returns true, and propery names corresponding to the
  509.  * argument names used in the format spec will be created on the event object.
  510.  * All optional parameters will be initialized to |null| if not already present
  511.  * on the event.
  512.  *
  513.  * On failure this method returns false and a description of the problem
  514.  * will be stored in the |parseError| property of the event.
  515.  *
  516.  * For example...
  517.  * Given the argument spec "<int> <word> [ <word2> <word3> ]", and given the
  518.  * input string "411 foo", stored as |e.command.usage| and |e.inputData|
  519.  * respectively, this method would add the following propertys to the event
  520.  * object...
  521.  *   -name---value--notes-   
  522.  *   e.int    411   Parsed as an integer
  523.  *   e.word   foo   Parsed as a string
  524.  *   e.word2  null  Optional parameters not specified will be set to null.
  525.  *   e.word3  null  If word2 had been provided, word3 would be required too.
  526.  *
  527.  * Each parameter is parsed by calling the function with the same name, located
  528.  * in this.argTypes.  The first parameter is parsed by calling the function
  529.  * this.argTypes["int"], for example.  This function is expected to act on
  530.  * e.unparsedData, taking it's chunk, and leaving the rest of the string.
  531.  * The default parse functions are...
  532.  *   <word>    parses contiguous non-space characters.
  533.  *   <int>     parses as an int.
  534.  *   <rest>    parses to the end of input data.
  535.  *   <state>   parses yes, on, true, 1, 0, false, off, no as a boolean.
  536.  *   <toggle>  parses like a <state>, except allows "toggle" as well.
  537.  *   <...>     parses according to the parameter type before it, until the end
  538.  *             of the input data.  Results are stored in an array named
  539.  *             paramnameList, where paramname is the name of the parameter
  540.  *             before <...>.  The value of the parameter before this will be
  541.  *             paramnameList[0].
  542.  *
  543.  * If there is no parse function for an argument type, "word" will be used by
  544.  * default.  You can alias argument types with code like...
  545.  * commandManager.argTypes["my-integer-name"] = commandManager.argTypes["int"];
  546.  */
  547. CommandManager.prototype.parseArgumentsRaw =
  548. function parse_parseargsraw (e)
  549. {
  550.     var argc = e.command.argNames.length;
  551.     
  552.     function initOptionals()
  553.     {
  554.         for (var i = 0; i < argc; ++i)
  555.         {
  556.             if (e.command.argNames[i] != ":" && 
  557.                 e.command.argNames[i] != "..."  && 
  558.                 !(e.command.argNames[i] in e))
  559.             {
  560.                 e[e.command.argNames[i]] = null;
  561.             }
  562.         }
  563.     }
  564.         
  565.     if ("inputData" in e && e.inputData)
  566.     {
  567.         /* if data has been provided, parse it */
  568.         e.unparsedData = e.inputData;
  569.         var parseResult;
  570.         var currentArg;
  571.         e.currentArgIndex = 0;
  572.  
  573.         if (argc)
  574.         {
  575.             currentArg = e.command.argNames[e.currentArgIndex];
  576.         
  577.             while (e.unparsedData)
  578.             {
  579.                 if (currentArg != ":")
  580.                 {
  581.                     if (!this.parseArgument (e, currentArg))
  582.                         return false;
  583.                 }
  584.                 if (++e.currentArgIndex < argc)
  585.                     currentArg = e.command.argNames[e.currentArgIndex];
  586.                 else
  587.                     break;
  588.             }
  589.  
  590.             if (e.currentArgIndex < argc && currentArg != ":")
  591.             {
  592.                 /* parse loop completed because it ran out of data.  We haven't
  593.                  * parsed all of the declared arguments, and we're not stopped
  594.                  * at an optional marker, so we must be missing something
  595.                  * required... */
  596.                 e.parseError = getMsg(MSN_ERR_REQUIRED_PARAM, 
  597.                                       e.command.argNames[e.currentArgIndex]);
  598.                 return false;
  599.             }
  600.         }
  601.         
  602.         if (e.unparsedData)
  603.         {
  604.             /* parse loop completed with unparsed data, which means we've
  605.              * successfully parsed all arguments declared.  Whine about the
  606.              * extra data... */
  607.             display (getMsg(MSN_EXTRA_PARAMS, e.unparsedData), MT_WARN);
  608.         }
  609.         else 
  610.         {
  611.             /* we've got no unparsed data, and we're not missing a required
  612.              * argument, go back and fill in |null| for any optional arguments
  613.              * not present on the comand line. */
  614.             initOptionals();
  615.         }
  616.     }
  617.     else
  618.     {
  619.         /* if no data was provided, check to see if the event is good enough
  620.          * on its own. */
  621.         var rv = this.isCommandSatisfied (e);
  622.         if (rv)
  623.             initOptionals();
  624.         return rv;
  625.     }
  626.  
  627.     return true;
  628. }
  629.  
  630. /**
  631.  * Returns true if |e| has the properties required to call the command
  632.  * |command|.
  633.  *
  634.  * If |command| is not provided, |e.command| is used instead.
  635.  *
  636.  * @param e        Event object to test against the command.
  637.  * @param command  Command to test.
  638.  */
  639. CommandManager.prototype.isCommandSatisfied =
  640. function cmgr_isok (e, command)
  641. {
  642.     if (typeof command == "undefined")
  643.         command = e.command;
  644.     
  645.     if (!command.enabled)
  646.         return false;
  647.  
  648.     for (var i = 0; i < command.argNames.length; ++i)
  649.     {
  650.         if (command.argNames[i] == ":")
  651.             return true;
  652.         
  653.         if (!(command.argNames[i] in e))
  654.         {
  655.             e.parseError = getMsg(MSN_ERR_REQUIRED_PARAM, command.argNames[i]);
  656.             //dd("command '" + command.name + "' unsatisfied: " + e.parseError);
  657.             return false;
  658.         }
  659.     }
  660.  
  661.     //dd ("command '" + command.name + "' satisfied.");
  662.     return true;
  663. }
  664.  
  665. /**
  666.  * Internal use only.
  667.  * See parseArguments above and the |argTypes| object below.
  668.  *
  669.  * Parses the next argument by calling an appropriate parser function, or the
  670.  * generic "word" parser if none other is found.
  671.  *
  672.  * @param e     event object.
  673.  * @param name  property name to use for the parse result.
  674.  */
  675. CommandManager.prototype.parseArgument =
  676. function cmgr_parsearg (e, name)
  677. {
  678.     var parseResult;
  679.     
  680.     if (name in this.argTypes)
  681.         parseResult = this.argTypes[name](e, name, this);
  682.     else
  683.         parseResult = this.argTypes["word"](e, name, this);
  684.  
  685.     if (!parseResult)
  686.         e.parseError = getMsg(MSN_ERR_INVALID_PARAM,
  687.                               [name, e.unparsedData]);
  688.  
  689.     return parseResult;
  690. }
  691.  
  692. CommandManager.prototype.argTypes = new Object();
  693.  
  694. /**
  695.  * Convenience function used to map a list of new types to an existing parse
  696.  * function.
  697.  */
  698. CommandManager.prototype.argTypes.__aliasTypes__ =
  699. function at_alias (list, type)
  700. {
  701.     for (var i in list)
  702.     {
  703.         this[list[i]] = this[type];
  704.     }
  705. }
  706.  
  707. /**
  708.  * Internal use only.
  709.  *
  710.  * Parses an integer, stores result in |e[name]|.
  711.  */
  712. CommandManager.prototype.argTypes["int"] =
  713. function parse_int (e, name)
  714. {
  715.     var ary = e.unparsedData.match (/(\d+)(?:\s+(.*))?$/);
  716.     if (!ary)
  717.         return false;
  718.     e[name] = Number(ary[1]);
  719.     e.unparsedData = arrayHasElementAt(ary, 2) ? ary[2] : "";
  720.     return true;
  721. }
  722.  
  723. /**
  724.  * Internal use only.
  725.  *
  726.  * Parses a word, which is defined as a list of nonspace characters.
  727.  *
  728.  * Stores result in |e[name]|.
  729.  */
  730. CommandManager.prototype.argTypes["word"] =
  731. function parse_word (e, name)
  732. {
  733.     var ary = e.unparsedData.match (/(\S+)(?:\s+(.*))?$/);
  734.     if (!ary)
  735.         return false;
  736.     e[name] = ary[1];
  737.     e.unparsedData = arrayHasElementAt(ary, 2) ? ary[2] : "";
  738.     return true;
  739. }
  740.  
  741. /**
  742.  * Internal use only.
  743.  *
  744.  * Parses a "state" which can be "true", "on", "yes", or 1 to indicate |true|,
  745.  * or "false", "off", "no", or 0 to indicate |false|.
  746.  *
  747.  * Stores result in |e[name]|.
  748.  */
  749. CommandManager.prototype.argTypes["state"] =
  750. function parse_state (e, name)
  751. {
  752.     var ary =
  753.         e.unparsedData.match (/(true|on|yes|1|false|off|no|0)(?:\s+(.*))?$/i);
  754.     if (!ary)
  755.         return false;
  756.     if (ary[1].search(/true|on|yes|1/i) != -1)
  757.         e[name] = true;
  758.     else
  759.         e[name] = false;
  760.     e.unparsedData = arrayHasElementAt(ary, 2) ? ary[2] : "";
  761.     return true;
  762. }
  763.  
  764. /**
  765.  * Internal use only.
  766.  *
  767.  * Parses a "toggle" which can be "true", "on", "yes", or 1 to indicate |true|,
  768.  * or "false", "off", "no", or 0 to indicate |false|.  In addition, the string
  769.  * "toggle" is accepted, in which case |e[name]| will be the string "toggle".
  770.  *
  771.  * Stores result in |e[name]|.
  772.  */
  773. CommandManager.prototype.argTypes["toggle"] =
  774. function parse_toggle (e, name)
  775. {
  776.     var ary = e.unparsedData.match
  777.         (/(toggle|true|on|yes|1|false|off|no|0)(?:\s+(.*))?$/i);
  778.  
  779.     if (!ary)
  780.         return false;
  781.     if (ary[1].search(/toggle/i) != -1)
  782.         e[name] = "toggle";
  783.     else if (ary[1].search(/true|on|yes|1/i) != -1)
  784.         e[name] = true;
  785.     else
  786.         e[name] = false;
  787.     e.unparsedData = arrayHasElementAt(ary, 2) ? ary[2] : "";
  788.     return true;
  789. }
  790.  
  791. /**
  792.  * Internal use only.
  793.  *
  794.  * Returns all unparsed data to the end of the line.
  795.  *
  796.  * Stores result in |e[name]|.
  797.  */
  798. CommandManager.prototype.argTypes["rest"] =
  799. function parse_rest (e, name)
  800. {
  801.     e[name] = e.unparsedData;
  802.     e.unparsedData = "";
  803.     return true;
  804. }
  805.  
  806. /**
  807.  * Internal use only.
  808.  *
  809.  * Parses the rest of the unparsed data the same way the previous argument was
  810.  * parsed.  Can't be used as the first parameter.  if |name| is "..." then the
  811.  * name of the previous argument, plus the suffix "List" will be used instead.
  812.  *
  813.  * Stores result in |e[name]| or |e[lastName + "List"]|.
  814.  */
  815. CommandManager.prototype.argTypes["..."] =
  816. function parse_repeat (e, name, cm)
  817. {
  818.     ASSERT (e.currentArgIndex > 0, "<...> can't be the first argument.");
  819.     
  820.     var lastArg = e.command.argNames[e.currentArgIndex - 1];
  821.     if (lastArg == ":")
  822.         lastArg = e.command.argNames[e.currentArgIndex - 2];
  823.  
  824.     var listName = lastArg + "List";
  825.     e[listName] = [e[lastArg]];
  826.     
  827.     while (e.unparsedData)
  828.     {
  829.         if (!cm.parseArgument(e, lastArg))
  830.             return false;
  831.         e[listName].push(e[lastArg]);
  832.     }    
  833.  
  834.     e[lastArg] = e[listName][0];
  835.     return true;
  836. }
  837.